-
Notifications
You must be signed in to change notification settings - Fork 377
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add AppSec::ActionHandler to unify action handling #4300
base: master
Are you sure you want to change the base?
Conversation
lib/datadog/appsec/action_handler.rb
Outdated
case type | ||
when 'block_request' then block_request(action_params) | ||
when 'redirect_request' then redirect_request(action_params) | ||
when 'generate_stack' then generate_stack(action_params) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when 'generate_stack' then generate_stack(action_params) | |
when 'generate_stack'generate_stack(action_params) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
lib/datadog/appsec/action_handler.rb
Outdated
when 'redirect_request' then redirect_request(action_params) | ||
when 'generate_stack' then generate_stack(action_params) | ||
when 'generate_schema' then generate_schema(action_params) | ||
when 'monitor' then monitor(action_params) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when 'monitor' then monitor(action_params) | |
when 'monitor'monitor(action_params) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
lib/datadog/appsec/action_handler.rb
Outdated
when 'block_request' then block_request(action_params) | ||
when 'redirect_request' then redirect_request(action_params) | ||
when 'generate_stack' then generate_stack(action_params) | ||
when 'generate_schema' then generate_schema(action_params) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when 'generate_schema' then generate_schema(action_params) | |
when 'generate_schema'generate_schema(action_params) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
lib/datadog/appsec/action_handler.rb
Outdated
|
||
def handle(type, action_params) | ||
case type | ||
when 'block_request' then block_request(action_params) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when 'block_request' then block_request(action_params) | |
when 'block_request'block_request(action_params) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
body << content(content_type) | ||
def block_response(action_params, http_accept_header) | ||
content_type = case action_params['type'] | ||
when nil, 'auto' then content_type(http_accept_header) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when nil, 'auto' then content_type(http_accept_header) | |
when nil, 'auto'content_type(http_accept_header) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
status: options['status_code']&.to_i || 403, | ||
headers: { 'Content-Type' => content_type }, | ||
body: body, | ||
status: (status_code >= 300 && status_code < 400 ? status_code : 303), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
Consider using ranges or between to simplify your comparison (...read more)
The rule "Prefer ranges/between over complex comparisons" advises developers to use the range or between?
method for comparisons instead of complex conditional statements. This practice increases the readability and clarity of your code. Complex comparisons using logical operators can be difficult to understand and prone to errors.
This rule is important because it promotes cleaner, more efficient, and easier-to-read code. When code is easier to read, it's easier to maintain, debug, and less likely to contain hidden bugs. Using the range or between?
method is a more concise way to check if a value falls within a specific range.
To adhere to this rule, replace complex comparison statements with the range or between?
method. For example, instead of writing foo >= 42 && foo <= 99
, you can write (42..99).include?(foo)
or foo.between?(42, 99)
. These alternatives are more straightforward and visually cleaner, making your code easier to understand.
lib/datadog/appsec/action_handler.rb
Outdated
def handle(type, action_params) | ||
case type | ||
when 'block_request' then block_request(action_params) | ||
when 'redirect_request' then redirect_request(action_params) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
⚪ Code Quality Violation
when 'redirect_request' then redirect_request(action_params) | |
when 'redirect_request'redirect_request(action_params) |
Don't use `then` for multi-line if, unless, when, or in statements (...read more)
The then
keyword is not necessary in multi-line if/unless/when/in
statements in Ruby. When used in multi-line statements, it can make the code harder to read and understand. This is because then
is typically associated with single-line conditional statements in Ruby, and its use in multi-line statements can be confusing.
Maintaining readability and clarity in your code is crucial for effective collaboration and debugging. It becomes even more important in larger codebases, where complex logic can become difficult to follow if not written clearly.
To avoid this issue, omit the then
keyword in your multi-line if/unless/when/in
statements. For single-line if/unless/when/in
statements, using then
is acceptable and can help improve readability. This practice keeps your code clean and easy to understand, following the principles of good coding practices.
|
||
Response.new( | ||
status: options['status_code']&.to_i || 403, | ||
headers: { 'Content-Type' => content_type }, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Content-Type
header is not needed for a redirect response
Datadog ReportBranch report: ✅ 0 Failed, 22066 Passed, 1477 Skipped, 5m 53.4s Total Time |
BenchmarksBenchmark execution time: 2025-01-17 16:17:16 Comparing candidate commit 4d47360 in PR branch Found 0 performance improvements and 0 performance regressions! Performance is the same for 31 metrics, 2 unstable metrics. |
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## master #4300 +/- ##
=======================================
Coverage 97.71% 97.71%
=======================================
Files 1358 1359 +1
Lines 82501 82478 -23
Branches 4223 4197 -26
=======================================
- Hits 80614 80593 -21
+ Misses 1887 1885 -2 ☔ View full report in Codecov by Sentry. |
5d7a7f9
to
3343f03
Compare
# Returns an error * the number of queries so that the entire multiplex is blocked | ||
if multiplex_response | ||
blocked_event = multiplex_response.find { |action, _options| action == :block } | ||
multiplex_return = AppSec::Response.graphql_response(gateway_multiplex) if blocked_event |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If I understand correctly, this PR also removes the creation of responses at framework level (e.g. GraphQL) to handle them all at Rack level ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yes, GraphQL instrumentation should only signal with INTERRUPT
and leave response blocking to Rack
end | ||
|
||
block = GraphQL::Reactive::Multiplex.publish(engine, gateway_multiplex) | ||
GraphQL::Reactive::Multiplex.publish(engine, gateway_multiplex) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In the files contained in the Reactive folder of each contrib (e.g. appsec/contrib/graphql/reactive/multiplex.rb
at the end of the subscribe
methods, we still throw :block
if any action is present, which is already wrong in itself. I wanted to fix it in the stacktrace PR but considering that this PR changes the way we handle actions, you may want to do it in this PR by deleting them ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This would be done in a separate PR to reduce the amount of changes here. We want to fully get rid of the Reactive Engine, since it only has a instrumentation-local scope and didn't work as intended
@@ -39,13 +38,15 @@ def watch_multiplex(gateway = Instrumentation.gateway) | |||
|
|||
Datadog::AppSec::Event.tag_and_keep!(context, result) | |||
context.events << event | |||
|
|||
result.actions.each do |action_type, action_params| |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So what if the actions hash looks like {"block"=>{...}, "generate_stack"=>{...}}
? Do we still want to generate the stacktrace ? If so, I suppose that we should do a first catch in this each
, save the block
state, handle all the other actions, then throw block
again
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
no, we don't need to throw block
, we want to generate the stack trace and then throw INTERRUPT
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
but you are right, we don't handle the action precedence correctly now, I will think about it and fix it
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
done 1a5404d
Because we need to handle actions according their precedence.
We want to test that ActionsHandler handles actions correctly according their precedence.
These examples have to be rewritten with Hash#merge, since older versions of Ruby do not support double-splat operator.
4c28ff6
to
4d47360
Compare
What does this PR do?
It adds
Datadog::AppSec::ActionHandler
module and unifies blocked response generation.https://github.com/DataDog/libddwaf/blob/master/UPGRADING.md#action-semantics
https://github.com/DataDog/appsec-diagrams/blob/main/action-handling.md
Motivation:
We want to have a simple and unified way of handling
libddwaf
actions.Change log entry
None.
Additional Notes:
Since Sinatra and Rails instrumentations always insert
Datadog::AppSec::Contrib::Rack::RequestMiddleware
, we don't need to catch interrupt signals anywhere other than in this Rack Middleware.How to test the change?
CI and manual testing with app generator.